package com.hyl.gulimall.order.config;

import org.springframework.amqp.core.Message;
import org.springframework.amqp.rabbit.connection.CorrelationData;
import org.springframework.amqp.rabbit.core.RabbitTemplate;
import org.springframework.amqp.support.converter.Jackson2JsonMessageConverter;
import org.springframework.amqp.support.converter.MessageConverter;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;

import javax.annotation.PostConstruct;

 @Configuration
public class MyRabbitConfig {

    @Autowired
    RabbitTemplate rabbitTemplate;

     /**
      * 使用JSON序列化机制，进行消息转换
      * @return
      */
    @Bean
    public MessageConverter messageConverter(){
        return new Jackson2JsonMessageConverter();
    }

    /**
     * 定制RabbitTemplate
     * 1.服务都到消息就回调
     *   1.spring.rabbitmq.publisher-confirms=true
     *   2.设置确认回调
     * 2.消息正确抵达队列进行回调
     *   1.spring.rabbitmq.publisher-returns=true
     *     spring.rabbitmq.template.mandatory=true
     *   2.设置确认回调ReturnCallback
     *
     * 3.消费端确认（保证每一个消息被正确消费，此时在可以broker删除这个消息）。
     * spring.rabbitmq.listener.direct.acknowledge-mode=manual 手动签收
     *  1.默认是自动确认的，只要消息接收到，客户端会自动确认，服务端就会移除这个消息
     *      问题：
     *          我们收到很多消息，自动回复给服务器ack，只有一个消息处理成功，宕机了，数据丢失
     *          消费者手动确认模式。只要我们没有明确告诉mq，货物背签收，没有ack，
     *          消息就一直Unacked状态。即使Consumer宕机。消息不会丢失，会重新变为Ready，下一次有新的Consumer连接进来就发给它
     *   2.如何签收
     *      channel.basicAck(deliveryTag,false);签收货物，业务成功完成就应该签收
     *      channel.basicNack(deliveryTag,false,true);拒签，业务失败，拒签
     */
    @PostConstruct  //MyRabbitConfig对象创建完成以后,执行这个方法
    public void initRabbitTemplate(){
        //确认回调
        rabbitTemplate.setConfirmCallback(new RabbitTemplate.ConfirmCallback() {
            /**
             * 1.只要消息抵达broker就ack=true
             * @param correlationData 当前消息的唯一关联数据（这个是消息的唯一id）
             * @param ack 消息是否成功
             * @param cause 失败的原因
             */
            @Override
            public void confirm(CorrelationData correlationData, boolean ack, String cause) {

                /**
                 * 1.做好消息确认机制（pulisher，consumer【手动ack】）
                 * 2、每一个发送的消息都在数据库做好记录，定期将失败的消息在次发送一遍
                 */

                //服务器收到了
                //修改消息的状态
                System.out.println("confirm...correlationData["+correlationData+"]==>ack["+ack+"]===>cause["+cause+"]");
            }
        });

        //设置消息毒打队列的确认回调,
        rabbitTemplate.setReturnCallback(new RabbitTemplate.ReturnCallback() {
            /**
             * 只要消息没有投递给指定的队列，就触发这个失败回调
             * @param message   投递失败的详细信息
             * @param i     回复的状态码
             * @param s     回复的文本内容
             * @param s1    当时这个消息发送给那个交换机
             * @param s2    当时这个消息发送给那个路由键
             */
            @Override
            public void returnedMessage(Message message, int i, String s, String s1, String s2) {
                //报错误了。修改数据库当前消息的状态-》错误
                System.out.println("Fail Message["+message+"]==>i["+i+"]==>s["+s+"]==>s1["+s1+"]==>s2["+s2+"]");
            }
        });

    }
}
